<script lang="ts" setup>
import type { Arrayable } from '@vueuse/core';
import type { FlattenedItem } from 'radix-vue';

import type { ClassType, Recordable } from '#/typings';

import type { TreeProps } from './types';

import { onMounted, ref, watchEffect } from 'vue';

import { ChevronRight, IconifyIcon } from '#/icons';
import { cn, get } from '#/utils';

import { TreeItem, TreeRoot } from 'radix-vue';

import { Checkbox } from '../checkbox';
import { treePropsDefaults } from './types';

const props = withDefaults(defineProps<TreeProps>(), treePropsDefaults());

const emits = defineEmits<{
  expand: [value: FlattenedItem<Recordable<any>>];
  select: [value: FlattenedItem<Recordable<any>>];
}>();

interface InnerFlattenItem<T = Recordable<any>, P = number | string> {
  hasChildren: boolean;
  id: P;
  level: number;
  parentId: null | P;
  parents: P[];
  value: T;
}

function flatten<T = Recordable<any>, P = number | string>(
    items: T[],
    childrenField: string = 'children',
    level = 0,
    parentId: null | P = null,
    parents: P[] = [],
): InnerFlattenItem<T, P>[] {
  const result: InnerFlattenItem<T, P>[] = [];
  items.forEach((item) => {
    const children = get(item, childrenField) as Array<T>;
    const id = get(item, props.valueField) as P;
    const val: InnerFlattenItem<T, P> = {
      hasChildren: Array.isArray(children) && children.length > 0,
      id,
      level,
      parentId,
      parents: [...parents],
      value: item,
    };
    result.push(val);
    if (val.hasChildren)
      result.push(
          ...flatten(children, childrenField, level + 1, id, [...parents, id]),
      );
  });
  return result;
}

const flattenData = ref<Array<InnerFlattenItem>>([]);
const modelValue = defineModel<Arrayable<number | string>>();
const expanded = ref<Array<number | string>>(props.defaultExpandedKeys ?? []);

const treeValue = ref();

onMounted(() => {
  watchEffect(() => {
    flattenData.value = flatten(props.treeData, props.childrenField);
    updateTreeValue();
    if (
        props.defaultExpandedLevel !== undefined &&
        props.defaultExpandedLevel > 0
    )
      expandToLevel(props.defaultExpandedLevel);
  });
});

function getItemByValue(value: number | string) {
  return flattenData.value.find(
      (item) => get(item.value, props.valueField) === value,
  )?.value;
}

function updateTreeValue() {
  const val = modelValue.value;
  if (val === undefined) {
    treeValue.value = undefined;
  } else {
    if (Array.isArray(val)) {
      const filteredValues = val.filter((v) => {
        const item = getItemByValue(v);
        return item && !get(item, props.disabledField);
      });
      treeValue.value = filteredValues.map((v) => getItemByValue(v));

      if (filteredValues.length !== val.length) {
        modelValue.value = filteredValues;
      }
    } else {
      const item = getItemByValue(val);
      if (item && !get(item, props.disabledField)) {
        treeValue.value = item;
      } else {
        treeValue.value = undefined;
        modelValue.value = undefined;
      }
    }
  }
}

function updateModelValue(val: Arrayable<Recordable<any>>) {
  if (Array.isArray(val)) {
    const filteredVal = val.filter((v) => !get(v, props.disabledField));
    modelValue.value = filteredVal.map((v) => get(v, props.valueField));
  } else {
    if (val && !get(val, props.disabledField)) {
      modelValue.value = get(val, props.valueField);
    }
  }
}

function expandToLevel(level: number) {
  const keys: string[] = [];
  flattenData.value.forEach((item) => {
    if (item.level <= level - 1) {
      keys.push(get(item.value, props.valueField));
    }
  });
  expanded.value = keys;
}

function collapseNodes(value: Arrayable<number | string>) {
  const keys = new Set(Array.isArray(value) ? value : [value]);
  expanded.value = expanded.value.filter((key) => !keys.has(key));
}

function expandNodes(value: Arrayable<number | string>) {
  const keys = [...(Array.isArray(value) ? value : [value])];
  keys.forEach((key) => {
    if (expanded.value.includes(key)) return;
    const item = getItemByValue(key);
    if (item) {
      expanded.value.push(key);
    }
  });
}

function expandAll() {
  expanded.value = flattenData.value
      .filter((item) => item.hasChildren)
      .map((item) => get(item.value, props.valueField));
}

function collapseAll() {
  expanded.value = [];
}

function checkAll() {
  if (!props.multiple) return;
  modelValue.value = [
    ...new Set(
        flattenData.value
            .filter((item) => !get(item.value, props.disabledField))
            .map((item) => get(item.value, props.valueField)),
    ),
  ];
  updateTreeValue();
}

function unCheckAll() {
  if (!props.multiple) return;
  modelValue.value = [];
  updateTreeValue();
}

function isNodeDisabled(item: FlattenedItem<Recordable<any>>) {
  return props.disabled || get(item.value, props.disabledField);
}

function onToggle(item: FlattenedItem<Recordable<any>>) {
  emits('expand', item);
}
function onSelect(item: FlattenedItem<Recordable<any>>, isSelected: boolean) {
  if (isNodeDisabled(item)) {
    return;
  }

  if (
      !props.checkStrictly &&
      props.multiple &&
      props.autoCheckParent &&
      isSelected
  ) {
    flattenData.value
        .find((i) => {
          return (
              get(i.value, props.valueField) === get(item.value, props.valueField)
          );
        })
        ?.parents?.filter((item) => !get(item, props.disabledField))
        ?.forEach((p) => {
          if (Array.isArray(modelValue.value) && !modelValue.value.includes(p)) {
            modelValue.value.push(p);
          }
        });
  }
  if (
      !props.checkStrictly &&
      props.multiple &&
      props.autoCheckParent &&
      !isSelected
  ) {
    flattenData.value
        .find((i) => {
          return (
              get(i.value, props.valueField) === get(item.value, props.valueField)
          );
        })
        ?.parents?.filter((item) => !get(item, props.disabledField))
        ?.reverse()
        .forEach((p) => {
          const children = flattenData.value.filter((i) => {
            return (
                i.parents.length > 0 &&
                i.parents.includes(p) &&
                i.id !== item._id &&
                i.parentId === p
            );
          });
          if (Array.isArray(modelValue.value)) {
            const hasSelectedChild = children.some((child) =>
                (modelValue.value as unknown[]).includes(
                    get(child.value, props.valueField),
                ),
            );
            if (!hasSelectedChild) {
              const index = modelValue.value.indexOf(p);
              if (index !== -1) {
                modelValue.value.splice(index, 1);
              }
            }
          }
        });
  }
  updateTreeValue();
  emits('select', item);
}

defineExpose({
  collapseAll,
  collapseNodes,
  expandAll,
  expandNodes,
  checkAll,
  unCheckAll,
  expandToLevel,
  getItemByValue,
});
</script>
<template>
  <TreeRoot
      :get-key="(item) => get(item, valueField)"
      :get-children="(item) => get(item, childrenField)"
      :items="treeData"
      :model-value="treeValue"
      v-model:expanded="expanded as string[]"
      :default-expanded="defaultExpandedKeys as string[]"
      :propagate-select="!checkStrictly"
      :multiple="multiple"
      :disabled="disabled"
      :selection-behavior="allowClear || multiple ? 'toggle' : 'replace'"
      @update:model-value="updateModelValue"
      v-slot="{ flattenItems }"
      :class="
      cn(
        'text-blackA11 container select-none list-none rounded-lg text-sm font-medium',
        $attrs.class as unknown as ClassType,
        bordered ? 'border' : '',
      )
    "
  >
    <div
        :class="
        cn('my-0.5 flex w-full items-center p-1', bordered ? 'border-b' : '')
      "
        v-if="$slots.header"
    >
      <slot name="header"> </slot>
    </div>
    <div
        :class="
        cn('my-0.5 flex w-full items-center p-1', bordered ? 'border-b' : '')
      "
        v-if="treeData.length > 0"
    >
      <div
          class="flex size-5 flex-1 cursor-pointer items-center"
          @click="() => (expanded?.length > 0 ? collapseAll() : expandAll())"
      >
        <ChevronRight
            :class="{ 'rotate-90': expanded?.length > 0 }"
            class="text-foreground/80 hover:text-foreground size-4 cursor-pointer transition"
        />
        <Checkbox
            v-if="multiple"
            @click.stop
            @update:checked="(checked) => (checked ? checkAll() : unCheckAll())"
        />
      </div>
    </div>
    <TransitionGroup :name="transition ? 'fade' : ''">
      <TreeItem
          v-for="item in flattenItems"
          v-slot="{
          isExpanded,
          isSelected,
          isIndeterminate,
          handleSelect,
          handleToggle,
        }"
          :key="item._id"
          :style="{ 'margin-left': `${item.level - 1}rem` }"
          :class="
          cn('cursor-pointer', getNodeClass?.(item), {
            'data-[selected]:bg-accent': !multiple,
            'text-foreground/50 cursor-not-allowed': isNodeDisabled(item),
          })
        "
          v-bind="
          Object.assign(item.bind, {
            onfocus: isNodeDisabled(item) ? 'this.blur()' : undefined,
            disabled: isNodeDisabled(item),
          })
        "
          @select="
          (event: any) => {
            if (isNodeDisabled(item)) {
              event.preventDefault();
              event.stopPropagation();
              return;
            }
            if (event.detail.originalEvent.type === 'click') {
              event.preventDefault();
            }
            onSelect(item, event.detail.isSelected);
          }
        "
          @toggle="
          (event: any) => {
            if (event.detail.originalEvent.type === 'click') {
              event.preventDefault();
            }
            !isNodeDisabled(item) && onToggle(item);
          }
        "
          class="tree-node focus:ring-grass8 my-0.5 flex items-center rounded p-1 outline-none focus:ring-2"
      >
        <ChevronRight
            v-if="
            item.hasChildren &&
            Array.isArray(item.value[childrenField]) &&
            item.value[childrenField].length > 0
          "
            class="text-foreground/80 hover:text-foreground size-4 cursor-pointer transition"
            :class="{ 'rotate-90': isExpanded }"
            @click.stop="
            () => {
              handleToggle();
              onToggle(item);
            }
          "
        />
        <div v-else class="h-4 w-4"></div>
        <div class="flex items-center gap-1">
          <Checkbox
              v-if="multiple"
              :checked="isSelected && !isNodeDisabled(item)"
              :disabled="isNodeDisabled(item)"
              :indeterminate="isIndeterminate && !isNodeDisabled(item)"
              @click="
              (event: MouseEvent) => {
                if (isNodeDisabled(item)) {
                  event.preventDefault();
                  event.stopPropagation();
                  return;
                }
                handleSelect();
              }
            "
          />
          <div
              class="flex items-center gap-1"
              @click="
              (event: MouseEvent) => {
                if (isNodeDisabled(item)) {
                  event.preventDefault();
                  event.stopPropagation();
                  return;
                }
                handleSelect();
              }
            "
          >
            <slot name="node" v-bind="item">
              <IconifyIcon
                  class="size-4"
                  v-if="showIcon && get(item.value, iconField)"
                  :icon="get(item.value, iconField)"
              />
              {{ get(item.value, labelField) }}
            </slot>
          </div>
        </div>
        <div class="h-4 w-4"></div>
      </TreeItem>
    </TransitionGroup>
    <div
        :class="
        cn('my-0.5 flex w-full items-center p-1', bordered ? 'border-t' : '')
      "
        v-if="$slots.footer"
    >
      <slot name="footer"> </slot>
    </div>
  </TreeRoot>
</template>
<style lang="scss" scoped>
.container {
  position: relative;
  padding: 0;
  list-style-type: none;
}

.item {
  box-sizing: border-box;
  width: 100%;
  height: 30px;
  background-color: #f3f3f3;
  border: 1px solid #666;
}

/* 1. 声明过渡效果 */
.fade-move,
.fade-enter-active,
.fade-leave-active {
  transition: all 0.5s cubic-bezier(0.55, 0, 0.1, 1);
}

/* 2. 声明进入和离开的状态 */
.fade-enter-from,
.fade-leave-to {
  opacity: 0;
  transform: scaleY(0.01) translate(30px, 0);
}

/* 3. 确保离开的项目被移除出了布局流
      以便正确地计算移动时的动画效果。 */
.fade-leave-active {
  position: absolute;
}
</style>